博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
框架Django
阅读量:6914 次
发布时间:2019-06-27

本文共 16190 字,大约阅读时间需要 53 分钟。

一.基本命令

1.新建一个django project

django-admin.py startproject project_name特别是在windows上如果报错,尝试使用django-admin代替django-admin.py

2.新建app

新建app需要切换到项目目录project_name下,即 cd project_name然后:python manage.py startapp app_name或django-admin.py startapp app_name

 

3.创建数据库表或更改数据库表或字段

Django 1.7.1及以上1.创建更改的文件python manage.py makemigrations2.将生成的py文件应用到数据库python manage.py migrate旧版本Django1.6及以下python manage.py syncdb

这种方法可以在SQL等数据库中创建与models.py代码对应的表,不需要自己手动执行SQL

4.使用开发服务器

开发服务器,一般修改代码后会自动重启,方便调试和开发,但是由于性能问题,建议只用在测试,不可用在生产环境

python manage.py runserver IP:PORT

5.清空数据库

python manage.py flush

6.创建超级管理员

python manage.py create superuser修改用户密码python manage.py changepassword username

7.导出数据 导入数据

python manage.py dumpdata appname > appname.jsonpython manage.py loaddata appname.json

8.Django项目环境终端

python manage.py shell

如果安装了bpython或ipython会自动调用它们的界面

这个命令和直接运行python或bpython进入shell的区别:可以在这个shell中调用当前项目的models.py中的API,对于操作数据和小测试都非常方便

9.数据库命令行

python manage.py dbshell

Django会自动进入settings.py中设置的数据库,如果是MySQL或postgreSQL,则会要求输入数据库用户密码

10.更多命令

python manage.py

查看详细的命令列表

11.static静态文件配置

每个应用都需要配置static静态文件,在每个应用的目录下创建static文件夹,将每个应用所需的静态文件放在里面,在settings中加入STATIC_ROOT=(    os.path.join(BASE_DIR, "app_name/static"))
{% load staticfiles %}# 
 

 

二.视图与网址

Django中网址写在urls.py文件中,用正则表达式对应view.py中的一个函数或者generic类

2.1定义视图函数

 
from django.shortcuts import render
def index(request):    return render(request,'index.html')

定义一个index函数,第一个参数必须是request,与网页发来的请求有关,request变量中包含get或post的内容,用户浏览器,系统等信息在里面

2.2定义视图函数相关的URL网址,打开project_name/project_name/urls.py文件

from django.conf.urls import url,includefrom django.contrib import adminfrom blog import viewsurlpatterns = [    url(r'^admin/', admin.site.urls),    url(r'^$',views.index),    url(r'^index/',views.index),    # url(r'^(\d{4})/$',year_query) 如果想要获取匹配的值并传入后边的视图函数,可以把想要匹配的正则表达式加上括号,成为后边的视图函数的参数,并以位置参数的形式传入    # url(r'^(?P
\d{4})/(?P
\d{2})$',views.year_query), 命名传入参数,以关键字参数传入 # url(r'reg/',views.reg,name="register") url(r'^blog/',include('blog.urls')) #新的应用需要添加对应的urls避免耦合,通过import include,并以该种形式实现对应应用的路由]

路由分配系统

  功能:客服端访问的url的路径(path)与视图函数一一映射关系

  语法格式:

  urlpatterns = [

 

     url(正则表达式,views视图函数,参数,别名

    ]

  key: 通过路径分组传参数给视图函数

  向server端传参数方式:

  (1)通过数据:http://1277.0.0.1:8080/blog/?id=1200

  (2)通过路径:http://1277.0.0.1:8080/blog/1200

         url(r"blog/(\d{4})")

2.3URL中的name

url(r'^reg/',views.reg,name="register")

简单说,name可以用于在templates,models,views……中得到对应的网址,相当于给网址取了个名字,只要名字不变,网址变了也能通过名字获得

我们注册页面用的是/reg/,如果以后网址进行了变更,比如改成了/new_client/,但在网页中,代码中很多地方都是写死的/reg/,如果是这样把网站写死了,会在更改了网址(正则)后,模板(templates),视图(views.py用于跳转),模型(models.py,获取对象对应的地址)用了此网址的,都需要进行相应的更改,十分繁琐

如果网址修改成了/new_client/,即urls.py进行了更改获取的网址也会动态地跟着改变,但是后面的name='register'没有改变,这时{%url 'register' %}就会渲染对应的网址成/new_client/

三.快捷函数

def login(request):    if request.method=="POST":        username=request.POST.get("username")        pwd=request.POST.get("pwd")        if username=='xiaobai' and pwd=='123':            return redirect('/index/')    return render(request, "login.html")#总结: render和redirect的区别:#   1 if render的页面需要模板语言渲染,需要的将数据库的数据加载到html,那么所有的这一部分#     除了写在yuan_back的视图函数中,必须还要写在login中,代码重复,没有解耦.#   2 the most important: url没有跳转到/yuan_back/,而是还在/login/,所以当刷新后#     又得重新登录.

四. Template

使用大括号来引用变量

{

{ a.1 }}

{

{ a.0 }}


{
% for person in querySet %} {
{ person.name }}{
% endfor %}

{
% if i > 50 %} {
{ i }}{
%else %}

50

{
% endif %}==================def index(request): i=65 a=[11,22,33,44] d={
"name":"xiaobai","age":22} class Person(object): def __init__(self,name): self.name=name p1=Person("xiaobai") p2=Person("xiaohei") p3=Person("xiaoming") querySet=[p1,p2,p3] return render(request,"index.html",locals())

五.创建模型models

1.新建项目和应用django-admin.py startproject BookManagecd BookManagepython manage.py startapp Book2.添加应用将新建的应用Book添加到settings.py中的INSTALLED_APPS中INSTALLED_APPS = (    'django.contrib.admin',    'django.contrib.auth',    'django.contrib.contenttypes',    'django.contrib.sessions',    'django.contrib.messages',    'django.contrib.staticfiles',     'Book',)3.修改models.pyclass Book(models.Model):    title=models.CharField(max_length=32)    price=models.IntegerField()    date=models.DateField()    publish=models.CharField(max_length=32)    author=models.CharField(max_length=32)新建一个Book类,继承models.Model4.创建数据表,使用SQLite3cd BookManagepython manage.py makemigrationspython manage.py migrate

 六.标签的使用

{% if %}的使用

{% if i > 50 %}    {
{ i }}{
%else %}

50

{
% endif %} ==============
{% if %} 标签接受and,or或者not来测试多个变量值或者否定一个给定的变量{% if %} 标签不允许同一标签里同时出现and和or,否则逻辑容易产生歧义,例如下面的标签是不合法的:{% if obj1 and obj2 or obj3 %}

{% for %}的使用

    {
    % for obj in list %}
  • {
    { obj.name }}
  • {
    % endfor %}
#在标签里添加reversed来反序循环列表:    {
% for obj in list reversed %} ... {
% endfor %}#{% for %}标签可以嵌套: {
% for country in countries %}

{
{ country.name }}

    {
    % for city in country.city_list %}
  • {
    { city }}
  • {
    % endfor %}
{
% endfor %}#系统不支持中断循环,系统也不支持continue语句,{% for %}标签内置了一个forloop模板变量,#这个变量含有一些属性可以提供给你一些关于循环的信息1,forloop.counter表示循环的次数,它从1开始计数,第一次循环设为1: {
% for item in todo_list %}

{

{ forloop.counter }}: {
{ item }}

{
% endfor %}2,forloop.counter0 类似于forloop.counter,但它是从0开始计数,第一次循环设为03,forloop.revcounter4,forloop.revcounter05,forloop.first当第一次循环时值为True,在特别情况下很有用: {
% for object in objects %} {
% if forloop.first %}
  • {% else %}
  • {% endif %} {
    { object }}
  • {
    % endfor %} # 富有魔力的forloop变量只能在循环中得到,当模板解析器到达{% endfor %}时forloop就消失了# 如果你的模板context已经包含一个叫forloop的变量,Django会用{% for %}标签替代它# Django会在for标签的块中覆盖你定义的forloop变量的值# 在其他非循环的地方,你的forloop变量仍然可用#{% empty %}{
    {li }} {
    % for i in li %}
  • {
    { forloop.counter0 }}----{
    { i }}
  • {
    % empty %}
  • this is empty!
  • {
    % endfor %}# [11, 22, 33, 44, 55]# 0----11# 1----22# 2----33# 3----44# 4----55

    {% url %}引用配置的路由,与name方法连用

    6.1 加载标签库:自定义filter和simple_tag

    6.1.1在应用目录下创建templatetags目录

    6.1.2在templatetags目录下创建.py文件,如Mytag.py

    语法格式:      {
    {obj|filter:param}} # 1 add : 给变量加上相应的值 # # 2 addslashes : 给变量中的引号前加上斜线 # # 3 capfirst : 首字母大写 # # 4 cut : 从字符串中移除指定的字符 # # 5 date : 格式化日期字符串 # # 6 default : 如果值是False,就替换成设置的默认值,否则就是用本来的值 # # 7 default_if_none: 如果值是None,就替换成设置的默认值,否则就使用本来的值
    from django import templateregister=template.Library() #register名字是固定的,不能任意定义@register.filterdef multi(x,y):    return x * y@register.simple_tagdef multi_argu(x,y,z):    return x * y * z

    6.1.3在使用定义了simple_tag和filter的html文件中加载之前创建的Mytag.py:{% load Mytag%}

    6.1.3使用simple_tag和filter

    {
    { d|date:"Y-m-d" }}

    {
    { w|truncatechars:5 }}

    {
    { w|truncatewords:5 }}

    {
    { s|default:"此处为空" }}

    {
    { i|multi:5 }}

    {% multi_argu 4 5 6 %}

    =========================def index(request): i=10 w="sadasljdlasjdlajsldajsljalsjldjasljdalsjljalsjldajskjad" d=datetime.datetime.now() s="" return render(request,"index.html",locals())
    View Code

    七.模板继承

    在网站中,为了减少共用页面区域的重复和冗余代码,如导航,底部,访问统计代码等,Django提供了模板继承extend方法。模板集成就是先构造一个基础框架模板,而后在其子模板中对它所包含站点公用部分和定义块进行重载。

    下面是base.html 和 index.html

    "base.html"    
    Title
    • 菜单一
    • 菜单二
    • 菜单三
    {
    % block con %}

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    bai

    {
    % endblock %} {
    % block page %}

    PAGE

    {
    % endblock %}
    ======================================={
    % extends "base.html" %}{
    % block con %} {
    { block.super }} {
    # 继承con的所有内容并添加以下内容到con内容之后 #}

    xiao

    xiao

    xiao

    xiao

    xiao

    xiao

    xiao

    xiao

    {
    % endblock %}{
    % block page %}

    This is page

    {
    % endblock %}{
    % include "add.html" %}
    View Code

    八.数据库

    1    django默认支持sqlite,mysql, oracle,postgresql数据库。

          sqlite

                django默认使用sqlite的数据库,默认自带sqlite的数据库驱动 , 引擎名称:django.db.backends.sqlite3

          mysql

                引擎名称:django.db.backends.mysql

    2    mysql驱动程序

    •    MySQLdb(mysql python)
    •    mysqlclient
    •    MySQL
    •    PyMySQL(纯python的mysql驱动程序)

    3     在django的项目中会默认使用sqlite数据库,在settings里有如下设置:

    1    django默认支持sqlite,mysql, oracle,postgresql数据库。

         <1> sqlite

                django默认使用sqlite的数据库,默认自带sqlite的数据库驱动 , 引擎名称:django.db.backends.sqlite3

         <2> mysql

                引擎名称:django.db.backends.mysql

    2    mysql驱动程序

    •    MySQLdb(mysql python)
    •    mysqlclient
    •    MySQL
    •    PyMySQL(纯python的mysql驱动程序)

    3     在django的项目中会默认使用sqlite数据库,在settings里有如下设置:

    更改MySQL配置修改如下

    DATABASES = {    'default': {        'ENGINE': 'django.db.backends.mysql',         'NAME': 'books',    #你的数据库名称        'USER': 'root',   #你的数据库用户名        'PASSWORD': '', #你的数据库密码        'HOST': '', #你的数据库主机,留空默认为localhost        'PORT': '3306', #你的数据库端口    }}

    注意:

    NAME即数据库的名字,在mysql连接前该数据库必须已经创建,而上面的sqlite数据库下的db.sqlite3则是项目自动创建USER和PASSWORD分别是数据库的用户名和密码。设置完后,再启动我们的Django项目前,我们需要激活我们的mysql。然后,启动项目,会报错:no module named MySQLdb这是因为django默认你导入的驱动是MySQLdb,可是MySQLdb对于py3有很大问题,所以我们需要的驱动是PyMySQL所以,我们只需要找到项目名文件下的__init__,在里面写入:import pymysqlpymysql.install_as_MySQLdb()

    九.ORM模型

    模型之间的三种关系:一对一,一对多,多对多

      一对一:在主外键的关系基础上,给外键添加一个UNIQUE=True的属性

      一对多:就是主外键关系 foreign key

      多对多:(ManyToManyField)自动创建第三张表,也可以自己创建第三张表

    9.1 ORM 增加表记录

    from app01.models import *    #create方式一:   Author.objects.create(name='Alvin')    #create方式二:   Author.objects.create(**{"name":"alex"})    #save方式一:     author=Author(name="alvin")                            author.save()    #save方式二:     author=Author()                            author.name="alvin"                            author.save()

    一对多(ForeignKey):

    # 一对多的记录创建方式    # 方法一    # Book.objects.create(    #     title="小白的帅气是如何炼成的",    #     price=999.99,    #     publish_id=1    #    # )    # 方法二    # publish_obj=Publish.objects.get(id=2)    # Book.objects.create(    #     title="小白的英俊是如何炼成的",    #     price=999.99,    #     publish=publish_obj    #    # )

    多对多 (ManyToManyField)

    author1=Author.objects.get(id=1)    author2=Author.objects.filter(name='alvin')[0]    book=Book.objects.get(id=1)    book.authors.add(author1,author2)    #等同于:    book.authors.add(*[author1,author2])    book.authors.remove(*[author1,author2])    #-------------------    book=models.Book.objects.filter(id__gt=1)    authors=models.Author.objects.filter(id=1)[0]    authors.book_set.add(*book)    authors.book_set.remove(*book)    #-------------------    book.authors.add(1)    book.authors.remove(1)    authors.book_set.add(1)    authors.book_set.remove(1)#注意: 如果第三张表是通过models.ManyToManyField()自动创建的,那么绑定关系只有上面一种方式#     如果第三张表是自己创建的:class Book2Author(models.Model):      author=models.ForeignKey("Author")      Book=  models.ForeignKey("Book")#     那么就还有一种方式:            author_obj=models.Author.objects.filter(id=2).first()            book_obj  =models.Book.objects.filter(id=3).first()            s=models.Book2Author.objects.create(author_id=1,Book_id=2)            s.save()            s=models.Book2Author(author=author_obj,Book_id=1)            s.save()

    9.2 ORM 删除表记录

    Book.objects.filter(id=1).delete()

    9.3 ORM修改表记录

    Book.objects.filter(id=3).update(title="PHP")#--------------- save方法会将所有属性重新设定一遍,效率低-----------    obj=Book.objects.filter(id=3)[0]    obj.title="Python"    obj.save()

    想要显示对应的SQL语句,需要在settings加上日志记录配置

    LOGGING = {    'version': 1,    'disable_existing_loggers': False,    'handlers': {        'console':{            'level':'DEBUG',            'class':'logging.StreamHandler',        },    },    'loggers': {        'django.db.backends': {            'handlers': ['console'],            'propagate': True,            'level':'DEBUG',        },    }}
    View Code

    9.4 ORM查询表记录

    # 查询相关API:#  <1>filter(**kwargs):      它包含了与所给筛选条件相匹配的对象#  <2>all():                 查询所有结果#  <3>get(**kwargs):         返回与所给筛选条件相匹配的对象,返回结果有且只有一个,如果符合筛选条件的对象超过一个或者没有都会抛出错误。#-----------下面的方法都是对查询的结果再进行处理:比如 objects.filter.values()--------#  <4>values(*field):        返回一个ValueQuerySet——一个特殊的QuerySet,运行后得到的并不是一系列 model的实例化对象,而是一个可迭代的字典序列                                     #  <5>exclude(**kwargs):     它包含了与所给筛选条件不匹配的对象#  <6>order_by(*field):      对查询结果排序#  <7>reverse():             对查询结果反向排序#  <8>distinct():            从返回结果中剔除重复纪录#  <9>values_list(*field):   它与values()非常相似,它返回的是一个元组序列,values返回的是一个字典序列#  <10>count():              返回数据库中匹配查询(QuerySet)的对象数量。#  <11>first():               返回第一条记录#  <12>last():                返回最后一条记录#  <13>exists():             如果QuerySet包含数据,就返回True,否则返回False
    # ret=Book.objects.values("title") #
    # print(ret) # ret=Book.objects.values_list("title","price") #
    # print(ret) # ret=Book.objects.exclude(id=1) # print(ret) # book_list=Book.objects.all().iterator() #将查找对象转换为迭代器,节省资源 # for book in book_lis
     
    单表 的 表记录查询#    models.Tb1.objects.filter(id__lt=10, id__gt=1)   # 获取id大于1 且 小于10的值##    models.Tb1.objects.filter(id__in=[11, 22, 33])   # 获取id等于11、22、33的数据#    models.Tb1.objects.exclude(id__in=[11, 22, 33])  # not in##    models.Tb1.objects.filter(name__contains="ven")#    models.Tb1.objects.filter(name__icontains="ven") # icontains大小写不敏感##    models.Tb1.objects.filter(id__range=[1, 2])   # 范围bettwen and##    startswith,istartswith, endswith, iendswith,
    View Code

     9.5 单表正向查询和反向查询

    # 正向查找    # book_obj=Book.objects.filter(id=1).first()    # print(book_obj.publish.name) #人民出版社    # book_obj=Book.objects.filter(id=2).first()    # author_list=book_obj.authors.all()    # for author in author_list:    #     print(author.name)    # 反向查找    # pub=Publish.objects.get(name="人民出版社")    # ret=pub.book_set.all().values("title")    # print(ret)
    View Code

     9.6 多表条件关联正向和反向查询

    #正向查找#一对多ret=Book.objects.filter(title="python").values.("publish__addr")##[{'publish__addr': '北京'}]#查询书名python的出版社地址#多对多ret=Book.objects.filter(title="python").values("author__name")ret=Book.objects.filter(author__name="xiaobai").values("title")#查询书名为python的作者名字#反向查找ret=Author.objects.filter(book__title="python").values("name")
    View Code

    十.聚合查询和分组查询

    10.1 aggregate

    通过对QuerySet进行计算,返回一个聚合值的字典。aggregate()中每一个参数都指定一个包含在字典中的返回值。即在查询集上生成集合

    from django.db.models import Avg,Min,Sum,MaxBook.objects.all().aggregate(Avg("price"))#查询所有图书的平均价格aggregate()是QuerySet的一个终止子句,意思是说,它返回一个包含一些键值对的字典。键的名称是聚合值得标识符,值是计算而来的聚合值。键的名称是按照字段和聚合函数的名称自动生成出来的。如果想要指定一个名称,可以向聚合子句提供:Book.objects.aggregate(average_price=Avg("price")

    10.2 annotate

    通过计算查询结果中每一个对象所关联的对象集合,从而计算出总值,即为查询集的每一项生成聚合

    Book.objects.filter(author__name="xiaobai").values("title")#查询作为为xiaobai写的所有书Book.objects.filter(authors__name="xiaobai").aggregate(Sum("price"))#查询xiaobai所有书的价格#查询每个作者所写书的总价格涉及到分组Book.objects.values("author__name"").annotate(Sum("price"))#查询各个出版社价格最低的书Book.objects.vaues("publish__name").annotate(Min("price"))#[{"publish__name":"北大出版社","price__min":Decimal("10.00")},{
    "publish__name":"人民出版社","price__min":Decimal("20.00")},{
    "publish__name":"河北出版社","price__min":Decimal("30.00")},]

    10.3 F查询和Q查询

    from django.db.models import F,Q#F使用查询条件的值,专门取对象中某列值的操作Book.objects.all().update(price=F("price")+20)#Q构建检索条件Book.objects.filter(Q(price__gt=200) | Q(id__gt=5) , Q(title__startswith="小"))以小开头的价格或者ID大于5的书#Q可以对关键字参数进行封装,从而应用多个条件查询# 可以组合使用&,|操作符,当一个操作符是用于两个Q的对象,它产生一个新的Q对象。    Q(title__startswith='P') | Q(title__startswith='J') # Q对象可以用~操作符放在前面表示否定,也可允许否定与不否定形式的组合    Q(title__startswith='P') | ~Q(pub_date__year=2005)

     

    转载于:https://www.cnblogs.com/c491873412/p/7413415.html

    你可能感兴趣的文章
    微软推Azure区块链简化企业公有链
    查看>>
    配置Tomcat及日志
    查看>>
    zabbix配置(上)
    查看>>
    2019年移动社交APP竞品前瞻性分析
    查看>>
    MySQL-MMM高可用
    查看>>
    安装 PrestaShop 1.6 - 关于快速安装指南
    查看>>
    杂谈 --- 出去走走,开拓视野。
    查看>>
    filebeat 接入kafka
    查看>>
    事情越多越需要一款适合的任务管理软件
    查看>>
    iOS使用[核心动画]和[定时器]两种方式实现【呼吸灯动画】(仿蘑菇街价格标签)...
    查看>>
    iOS基础问答面试题连载(二)
    查看>>
    无密码FTP登录
    查看>>
    网站和短信设备相结合如何办到
    查看>>
    使用 OpenGL 遇到的 Bug - 'color undeclared'
    查看>>
    C++编写的十进制转换成16进制源码
    查看>>
    如何对CAD绘图软件中的页面进行设置
    查看>>
    一致性hash算法及其java实现!
    查看>>
    大数据教程(3.7):zookeeper集群自动化启动、关闭、重启脚本
    查看>>
    大数据教程(6.4)centos6.9安装hadoop2.9.1
    查看>>
    awk基础简介
    查看>>